Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Proof of concept non-component-based css prop. #84

Merged
merged 22 commits into from
Jan 24, 2025
Merged

Conversation

stephenh
Copy link
Contributor

@stephenh stephenh commented Nov 9, 2022

This is a spike related to https://dev.to/srmagura/why-were-breaking-up-wiht-css-in-js-4g9b and specifically re "can we get rid of the wrapping EmotionCssPropInternal components" that currently are required anytime a css= prop is used.

See reactwg/react-18#110 but it would be great if React had a non-hook way of subscribing to useInsertionEffect events, like document.addEventElement('insertionEffect", () => ...).

Misc ideas:

  1. Just try stock fela & FelaComponent, to see perf vs. emotion
    • Would still have wrapper components in the React tree, but just curious
  2. Call the fela renderer directly inline in __jsxDEV (no FelaComponent) and 🤷 if the performance of "not using useInsertionEffect really matters
    • I.e. styles would be injected ~randomly after each render and not necessarily at "safe insertion points", which would cause more DOM/CSS recalcs per the React-18 link
    • Complain to react about the lack of a global insertionEffect event/hook
  3. Output any non-selector styles directly to the style attribute, and skip class names all together (recommended in the react-18 doc as what FB does, but technically only for dynamic styles)
    • Output any selector-based styles via Emotion/Fela as usual
  4. Use already-injected awareness to only conditionally wrap in a FelaComponent, otherwise use the existing/injected classes directly on className (no FelaComponent)

@stephenh stephenh changed the title Proof of concept non-component-based css prop. feat: Proof of concept non-component-based css prop. Nov 9, 2022
@stephenh
Copy link
Contributor Author

stephenh commented Jan 18, 2023

Another idea for component-less usage:

export function FooComponent() {
  const css = useTruss();

  return <div className={css.mt2.mb3.$} />;
}
  • Pro: "Fine, fine" uses a hook to get css, which will play nicer with React 18's useInsertionEffect hook
  • Con: Burns a LOC in basically every component
  • Pro: Uses just className so doesn't need the css=... JSX wrapper
    • This removes the jsxImportSource / jsxFactory build config from vite/storybook/etc (which we have working now, but can be flakey when upgrading/changing configs)
    • This also removes the EmotionalInternalCss wrapper component that Emotion's css= prop is forced to inject for it to add its own hooks to resolve emotion from a context

@stephenh stephenh merged commit 35f420e into main Jan 24, 2025
2 checks passed
@stephenh stephenh deleted the fela-css-prop branch January 24, 2025 16:31
homebound-team-bot pushed a commit that referenced this pull request Jan 24, 2025
# [1.137.0](v1.136.0...v1.137.0) (2025-01-24)

### Features

* Proof of concept non-component-based css prop. ([#84](#84)) ([35f420e](35f420e))
@homebound-team-bot
Copy link

🎉 This PR is included in version 1.137.0 🎉

The release is available on:

Your semantic-release bot 📦🚀

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants